/*
 *  Licensed to the Apache Software Foundation (ASF) under one
 *  or more contributor license agreements.  See the NOTICE file
 *  distributed with this work for additional information
 *  regarding copyright ownership.  The ASF licenses this file
 *  to you under the Apache License, Version 2.0 (the
 *  "License"); you may not use this file except in compliance
 *  with the License.  You may obtain a copy of the License at
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 *  Unless required by applicable law or agreed to in writing,
 *  software distributed under the License is distributed on an
 *  "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 *  KIND, either express or implied.  See the License for the
 *  specific language governing permissions and limitations
 *  under the License.
 */
package org.codehaus.groovy.classgen

def ops = [
        'plus',
        'minus',
        'multiply',
        'div',
        'or',
        'and',
        'xor',
        'intdiv',
        'remainder',
        'leftShift',
        'rightShift',
        'rightShiftUnsigned'
]

def numbers = ['Byte':'byte', 'Short':'short', 'Integer':'int', 'Long':'long', 'Float':'float', 'Double':'double']

ops.each { op ->
    numbers.each { wrappedType, type ->
        println "public boolean ${type}_${op};"
    }
}

ops.each { op ->
    println "if (\"${op}\".equals(name)) {"
    numbers.each { wrappedType, type ->
        println """if (klazz==${wrappedType}.class) {
                ${type}_${op} = true;
            }"""
    }
    println 'if (klazz==Object.class) {'
    numbers.each { wrappedType, type ->
        println "${type}_${op} = true;"
            }
    println '}'
    println '}'
}

ops.each { op ->
    numbers.each { wrappedType1, type1 ->
        numbers.each { wrappedType2, type2 ->
            def math = getMath(wrappedType1, wrappedType2)
            if (math [op]) {
                println """public static ${math.resType} ${op}(${type1} op1, ${type2} op2) {
                   if (instance.${type1}_${op}) {
                      return ${op}Slow(op1, op2);
                   }
                   else {
                      return ${math.resType != type1 ? '((' + math.resType+ ')op1)' : 'op1'} ${math[op]} ${math.resType != type2 ? '((' + math.resType+ ')op2)' : 'op2'};
                   }
                }"""
                println """private static ${math.resType} ${op}Slow(${type1} op1,${type2} op2) {
                      return ((Number)InvokerHelper.invokeMethod(op1, "${op}", op2)).${math.resType}Value();
                }"""
            }
        }
    }
}

def isFloatingPoint(number) {
    number == 'Double' || number == 'Float'
}

def isLong(number) {
    number == 'Long'
}

def getMath (left, right) {
    if (isFloatingPoint(left) || isFloatingPoint(right)) {
        return [
                resType : 'double',

                plus : '+',
                minus : '-',
                multiply : '*',
                div : '/',
        ]
    }
    if (isLong(left) || isLong(right)){
        return [
                resType : 'long',

                plus : '+',
                minus : '-',
                multiply : '*',
                div : '/',
                or : '|',
                and : '&',
                xor : '^',
                intdiv : '/',
                remainder : '%',
                leftShift : '<<',
                rightShift : '>>',
                rightShiftUnsigned : '>>>'
        ]
    }
    [
            resType : 'int',

            plus : '+',
            minus : '-',
            multiply : '*',
            div : '/',
            or : '|',
            and : '&',
            xor : '^',
            intdiv : '/',
            remainder : '%',
            leftShift : '<<',
            rightShift : '>>',
            rightShiftUnsigned : '>>>'
    ]
}
